Security News
Node.js EOL Versions CVE Dubbed the "Worst CVE of the Year" by Security Experts
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
@travetto/di
Advanced tools
Install: primary
$ npm install @travetto/di
Dependency injection
is a framework primitive. When used in conjunction with automatic file scanning, it provides for handling of application dependency wiring. Due to the nature of typescript
and type erasure of interfaces, dependency injection only supports class
es as type signafiers. The primary goal of dependency injection is to allow for separation of concerns of object creation and it's usage.
The @Injectable
and @InjectableFactory
decorators provide the registration of dependencies. Dependency declaration revolves around exposing class
es and subtypes thereof to provide necessary functionality. Additionally, the framework will utilize dependencies to satisfy contracts with various backends (e.g. MongoModelSource
provides itself as an injectable candidate for ModelSource
).
Code: Example @Injectable
@Injectable()
class CustomService {
async coolOperation() {
... work ...
}
}
When declaring a dependency, you can also provide a token to allow for multiple instances of the dependency to be defined. This can be used in many situations:
Code: Example @Injectable with multiple targets
@Injectable()
class CustomService {
async coolOperation() {
... work ...
}
}
const CUSTOM2 = Symbol('custom2');
@Injectable({ target: CustomService, symbol: CUSTOM2 })
class CustomService2 extends CustomService {
async coolOperation() {
await super.coolOperation();
// Do some additional work
}
}
class Consumer {
@Inject(CUSTOM2) // Pull in specific service
service: CustomService;
}
As you can see, the target
field is also set, which indicates to the dependency registration process what class
the injectable is compatible with. Additionally, when using abstract
classes, the parent class
is always considered as a valid candidate type.
Code: Example @Injectable with target via abstract class
abstract class BaseService {
abstract work():Promise<void>;
}
@Injectable()
class SpecificService extends BaseService {
async work() {
// Do some additional work
}
}
In this scenario, SpecificService
is a valid candidate for BaseService
due to the abstract inheritance. Sometimes, you may want to provide a slight variation to a dependency without extending a class. To this end, the @InjectableFactory
decorator denotes a static
class method that produces an @Injectable
.
Code: Example @InjectableFactory, return type defines target class
class Config {
@InjectableFactory()
static initService(): CoolService {
return new CoolService();
}
}
Given the static
method initService
, the function will be provided as a valid candidate for CoolService
. Instead of calling the constructor of the type directly, this function will work as a factory for producing the injectable.
NOTE Due to the lack of typechecker in the Compiler
for performance reasons, the return type on the factory method is mandatory. Without it, the code will not know what the expected target type should be.
NOTE Other modules are able to provide aliases to @Injectable
that also provide additional functionality. For example, the @Config
or the @Controller
decorator registers the associated class as an injectable element.
Once all of your necessary dependencies are defined, now is the time to provide those @Injectable
instances to your code. There are three primary methods for injection:
The @Inject
decorator, which denotes a desire to inject a value directly. These will be set post construction.
Code: Example @Injectable with dependencies as @Inject fields
@Injectable()
class CustomService {
@Inject()
private dependentService: DependentService;
async coolOperation() {
await this.dependentService.doWork();
}
}
The @Injectable
constructor params, which will be provided as the instance is being constructed.
Code: Example @Injectable with dependencies in constructor
@Injectable()
class CustomService {
constructor (private dependentService: DependentService) {}
async coolOperation() {
await this.dependentService.doWork();
}
}
Via @InjectableFactory
params, which are comparable to constructor params
Code: Example @InjectableFactory with parameters as dependencies
class Config {
@InjectableFactory()
static initService(dependentService: DependentService): CustomService {
return new CustomService(dependentService);
}
}
NOTE If you are building modules for others to consume, often times it is desirable to have default implementations for contracts that can be overridden. The @Inject
decorator takes a flag of defaultIfMissing
and a class name, that it will use to satisfy the dependency if nothing else matches. This would look like:
Code: Example of default if missing
abstract class Contract {
}
@Injectable({ targets: SimpleContract })
class SimpleContract extends Contract {}
@Injectable({ targets: ComplexContract })
class ComplexContract extends Contract {}
@Injectable()
class ContractConsumer {
// Will default to SimpleContract if nothing else registered
@Inject({ defaultIfMissing: SimpleContract })
contract: Contract;
}
class Config {
// Complex will be marked as the available Contract
@InjectableFactory()
getContract(complex: ComplexContract): Contract {
return complex;
}
}
Some times you will need to lookup a dependency dynamically, or you want to control the injection process at a more granular level. To achieve that you will need to directly access the DependencyRegistry
. The registry allows for requesting a dependency by class reference:
Code: Example of manual lookup
@Injectable()
class Complex {}
class ManualLookup {
async invoke() {
const complex = await DependencyRegistry.getInstance(Complex);
}
}
@Application
Given that dependency injection is generally a pre-requisite for application execution, it stands as the primary entrypoint for application invocation. The Base
provides a simplistic bootstrap to allow for the application to run, but that is not sufficient for more complex applications.
The module provides a decorator, @Application
who's job is to register entry points into the application. For example:
Code: Example of @Application target
import { Application, Inject, Injectable } from '../';
@Injectable()
class Server {
name = 'roger';
async launch() {
...
}
}
@Application('simple')
class SimpleApp {
@Inject()
server: Server
async run() {
return this.server.launch();
}
}
The @Application
decorator exposes some additional functionality, which can be used to launch the application.
.run()
ArgumentsThe arguments specified in the run
method, will now be able to be specified when invoking the application from the command line. For instance:
Code: Simple Entrypoint with Parameters
@Application('simple')
class SimpleApp {
async run(domain: string, port = 3000) {
console.log('Launching', domain, 'on port', port);
}
}
These command line invocation of travetto run
would look like:
Terminal: Sample CLI Output
$ travetto run
....
● [e2e] simple
----------------------------------
usage: simple [domain] (port=3000)
To invoke the simple
application, you need to pass domain
where port is optional with a default.
Terminal: Invoke Simple
$ travetto run simple mydomain.biz 4000
[INFO] Launching mydomain.biz on port 4000
Additionally, the parameters will be type checked, to ensure proper evaluation.
Terminal: Invoke Simple with bad port
$ travetto run simple mydomain.biz orange
usage: simple domain (string), port=[3000] (number)
The types are inferred from the .run()
method parameters, but can be overridden in the @Application
annotation to support customization. Only primitive types are supported:
number
- Float or decimalstring
- Default if no type is specifiedboolean
- true(yes/on/1) and false(no/off/0)union
- Type unions of the same type (string_a|string_b
or 1|2|3|4
)Customizing the types is done by name, and allows for greater control:
Code: Complex Entrypoint with Customization
@Application('complex', {
watchable: true,
paramMap: {
domain: {
title: 'Domain Name',
type: 'string',
subtype: 'url'
},
port : {
title: 'Server Port',
def: 3000
}
}
})
class ComplexApp {
async run(domain: string, port: number) {
console.log('Launching', domain, 'on port', port);
}
}
*** Note ***
The applications, by default, will not scan other application's folders. This means, if you have an application in the e2e/
folder, all of the code in your src/
folder will not be picked up automatically. This defined under the assumption that each application is unique. If you have an application that is an extension of the primary application (src/
), you can specify the @Application
config property of standalone
to be false. This will now scan both folders to run your application.
FAQs
Dependency registration/management and injection support.
The npm package @travetto/di receives a total of 614 weekly downloads. As such, @travetto/di popularity was classified as not popular.
We found that @travetto/di demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Critics call the Node.js EOL CVE a misuse of the system, sparking debate over CVE standards and the growing noise in vulnerability databases.
Security News
cURL and Go security teams are publicly rejecting CVSS as flawed for assessing vulnerabilities and are calling for more accurate, context-aware approaches.
Security News
Bun 1.2 enhances its JavaScript runtime with 90% Node.js compatibility, built-in S3 and Postgres support, HTML Imports, and faster, cloud-first performance.